/*
 *  Licensed to the Apache Software Foundation (ASF) under one or more
 *  contributor license agreements.  See the NOTICE file distributed with
 *  this work for additional information regarding copyright ownership.
 *  The ASF licenses this file to You under the Apache License, Version 2.0
 *  (the "License"); you may not use this file except in compliance with
 *  the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 *  Unless required by applicable law or agreed to in writing, software
 *  distributed under the License is distributed on an "AS IS" BASIS,
 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *  See the License for the specific language governing permissions and
 *  limitations under the License.
 */

package java.io;

/**
 * The base class for all writers. A writer is a means of writing data to a
 * target in a character-wise manner. Most output streams expect the
 * {@link #flush()} method to be called before closing the stream, to ensure all
 * data is actually written out.
 * <p>
 * This abstract class does not provide a fully working implementation, so it
 * needs to be subclassed, and at least the {@link #write(char[], int, int)},
 * {@link #close()} and {@link #flush()} methods needs to be overridden.
 * Overriding some of the non-abstract methods is also often advised, since it
 * might result in higher efficiency.
 * <p>
 * Many specialized readers for purposes like reading from a file already exist
 * in this package.
 *
 * @see Reader
 */
public abstract class Writer implements Appendable, Closeable, Flushable {
    /**
     * The object used to synchronize access to the writer.
     */
    protected Object lock;
    
    /**
     * Constructs a new {@code Writer} with {@code this} as the object used to
     * synchronize critical sections.
     */
    protected Writer() {
        lock = this;
    }
    
    /**
     * Constructs a new {@code Writer} with {@code lock} used to synchronize
     * critical sections.
     *
     * @param lock the {@code Object} used to synchronize critical sections.
     * @throws NullPointerException if {@code lock} is {@code null}.
     */
    protected Writer(Object lock) {
        if (lock == null) {
            throw new NullPointerException("lock == null");
        }
        this.lock = lock;
    }
    
    /**
     * Closes this writer. Implementations of this method should free any
     * resources associated with the writer.
     *
     * @throws IOException if an error occurs while closing this writer.
     */
    public abstract void close() throws IOException;
    
    /**
     * Flushes this writer. Implementations of this method should ensure that
     * all buffered characters are written to the target.
     *
     * @throws IOException if an error occurs while flushing this writer.
     */
    public abstract void flush() throws IOException;
    
    /**
     * Writes the entire character buffer {@code buf} to the target.
     *
     * @param buf the non-null array containing characters to write.
     * @throws IOException if this writer is closed or another I/O error occurs.
     */
    public void write(char[] buf) throws IOException {
        write(buf, 0, buf.length);
    }
    
    /**
     * Writes {@code count} characters starting at {@code offset} in {@code buf}
     * to the target.
     *
     * @param buf    the non-null character array to write.
     * @param offset the index of the first character in {@code buf} to write.
     * @param count  the maximum number of characters to write.
     * @throws IndexOutOfBoundsException if {@code offset < 0} or {@code count < 0}, or if {@code
     *                                   offset + count} is greater than the size of {@code buf}.
     * @throws IOException               if this writer is closed or another I/O error occurs.
     */
    public abstract void write(char[] buf, int offset, int count) throws IOException;
    
    /**
     * Writes one character to the target. Only the two least significant bytes
     * of the integer {@code oneChar} are written.
     *
     * @param oneChar the character to write to the target.
     * @throws IOException if this writer is closed or another I/O error occurs.
     */
    public void write(int oneChar) throws IOException {
        synchronized (lock) {
            char[] oneCharArray = new char[1];
            oneCharArray[0] = (char) oneChar;
            write(oneCharArray);
        }
    }
    
    /**
     * Writes the characters from the specified string to the target.
     *
     * @param str the non-null string containing the characters to write.
     * @throws IOException if this writer is closed or another I/O error occurs.
     */
    public void write(String str) throws IOException {
        write(str, 0, str.length());
    }
    
    /**
     * Writes {@code count} characters from {@code str} starting at {@code
     * offset} to the target.
     *
     * @param str    the non-null string containing the characters to write.
     * @param offset the index of the first character in {@code str} to write.
     * @param count  the number of characters from {@code str} to write.
     * @throws IOException               if this writer is closed or another I/O error occurs.
     * @throws IndexOutOfBoundsException if {@code offset < 0} or {@code count < 0}, or if {@code
     *                                   offset + count} is greater than the length of {@code str}.
     */
    public void write(String str, int offset, int count) throws IOException {
        if ((offset | count) < 0 || offset > str.length() - count) {
            throw new StringIndexOutOfBoundsException();
        }
        char[] buf = new char[count];
        str.getChars(offset, offset + count, buf, 0);
        synchronized (lock) {
            write(buf, 0, buf.length);
        }
    }
    
    /**
     * Appends the character {@code c} to the target. This method works the same
     * way as {@link #write(int)}.
     *
     * @param c the character to append to the target stream.
     * @return this writer.
     * @throws IOException if this writer is closed or another I/O error occurs.
     */
    public Writer append(char c) throws IOException {
        write(c);
        return this;
    }
    
    /**
     * Appends the character sequence {@code csq} to the target. This method
     * works the same way as {@code Writer.write(csq.toString())}. If {@code
     * csq} is {@code null}, then the string "null" is written to the target
     * stream.
     *
     * @param csq the character sequence appended to the target.
     * @return this writer.
     * @throws IOException if this writer is closed or another I/O error occurs.
     */
    public Writer append(CharSequence csq) throws IOException {
        if (csq == null) {
            csq = "null";
        }
        write(csq.toString());
        return this;
    }
    
    /**
     * Appends a subsequence of the character sequence {@code csq} to the
     * target. This method works the same way as {@code
     * Writer.writer(csq.subsequence(start, end).toString())}. If {@code
     * csq} is {@code null}, then the specified subsequence of the string "null"
     * will be written to the target.
     *
     * @param csq   the character sequence appended to the target.
     * @param start the index of the first char in the character sequence appended
     *              to the target.
     * @param end   the index of the character following the last character of the
     *              subsequence appended to the target.
     * @return this writer.
     * @throws IOException               if this writer is closed or another I/O error occurs.
     * @throws IndexOutOfBoundsException if {@code start > end}, {@code start < 0}, {@code end < 0} or
     *                                   either {@code start} or {@code end} are greater or equal than
     *                                   the length of {@code csq}.
     */
    public Writer append(CharSequence csq, int start, int end) throws IOException {
        if (csq == null) {
            csq = "null";
        }
        write(csq.subSequence(start, end).toString());
        return this;
    }
    
    /**
     * Returns true if this writer has encountered and suppressed an error. Used
     * by PrintWriters as an alternative to checked exceptions.
     */
    boolean checkError() {
        return false;
    }
}
